/*---------------------------------------------------------------------------*\
  =========                 |
  \\      /  F ield         | foam-extend: Open Source CFD
   \\    /   O peration     | Version:     4.1
    \\  /    A nd           | Web:         http://www.foam-extend.org
     \\/     M anipulation  | For copyright notice see file Copyright
-------------------------------------------------------------------------------
License
	This file is part of foam-extend.

	foam-extend is free software: you can redistribute it and/or modify it
	under the terms of the GNU General Public License as published by the
	Free Software Foundation, either version 3 of the License, or (at your
	option) any later version.

	foam-extend is distributed in the hope that it will be useful, but
	WITHOUT ANY WARRANTY; without even the implied warranty of
	MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
	General Public License for more details.

	You should have received a copy of the GNU General Public License
	along with foam-extend.  If not, see <http://www.gnu.org/licenses/>.

\*---------------------------------------------------------------------------*/

#include "GeometricFieldReuseFunctions.H"

#define TEMPLATE \
	template<class Type, template<class> class PatchField, class GeoMesh>
#include "GeometricFieldFunctionsM.C"

// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * //

namespace Foam
{

// * * * * * * * * * * * * * * * Global functions  * * * * * * * * * * * * * //

template<class Type, template<class> class PatchField, class GeoMesh>
void component
(
	GeometricField
	<
		typename GeometricField<Type, PatchField, GeoMesh>::cmptType,
		PatchField,
		GeoMesh
	>& gcf,
	const GeometricField<Type, PatchField, GeoMesh>& gf,
	const direction d
)
{
	component(gcf.internalField(), gf.internalField(), d);
	component(gcf.boundaryField(), gf.boundaryField(), d);
}


template<class Type, template<class> class PatchField, class GeoMesh>
void T
(
	 GeometricField<Type, PatchField, GeoMesh>& gf,
	 const GeometricField<Type, PatchField, GeoMesh>& gf1
)
{
	T(gf.internalField(), gf1.internalField());
	T(gf.boundaryField(), gf1.boundaryField());
}


template<class Type, template<class> class PatchField, class GeoMesh, int r>
void pow
(
	GeometricField<typename powProduct<Type, r>::type, PatchField, GeoMesh>& gf,
	const GeometricField<Type, PatchField, GeoMesh>& gf1
)
{
	pow(gf.internalField(), gf1.internalField(), r);
	pow(gf.boundaryField(), gf1.boundaryField(), r);
}

template<class Type, template<class> class PatchField, class GeoMesh, int r>
tmp<GeometricField<typename powProduct<Type, r>::type, PatchField, GeoMesh> >
pow
(
	const GeometricField<Type, PatchField, GeoMesh>& gf,
	typename powProduct<Type, r>::type
)
{
	typedef typename powProduct<Type, r>::type powProductType;

	tmp<GeometricField<powProductType, PatchField, GeoMesh> > tPow
	(
		new GeometricField<powProductType, PatchField, GeoMesh>
		(
			IOobject
			(
				"pow(" + gf.name() + ',' + name(r) + ')',
				gf.instance(),
				gf.db(),
				IOobject::NO_READ,
				IOobject::NO_WRITE
			),
			gf.mesh(),
			pow(gf.dimensions(), r)
		)
	);

	pow<Type, r, PatchField, GeoMesh>(tPow(), gf);

	return tPow;
}


template<class Type, template<class> class PatchField, class GeoMesh, int r>
tmp<GeometricField<typename powProduct<Type, r>::type, PatchField, GeoMesh> >
pow
(
	const tmp<GeometricField<Type, PatchField, GeoMesh> >& tgf,
	typename powProduct<Type, r>::type
)
{
	typedef typename powProduct<Type, r>::type powProductType;

	const GeometricField<Type, PatchField, GeoMesh>& gf = tgf();

	tmp<GeometricField<powProductType, PatchField, GeoMesh> > tPow
	(
		new GeometricField<powProductType, PatchField, GeoMesh>
		(
			IOobject
			(
				"pow(" + gf.name() + ',' + name(r) + ')',
				gf.instance(),
				gf.db(),
				IOobject::NO_READ,
				IOobject::NO_WRITE
			),
			gf.mesh(),
			pow(gf.dimensions(), r)
		)
	);

	pow<Type, r, PatchField, GeoMesh>(tPow(), gf);

	tgf.clear();

	return tPow;
}


template<class Type, template<class> class PatchField, class GeoMesh>
void sqr
(
	GeometricField
	<typename outerProduct<Type, Type>::type, PatchField, GeoMesh>& gf,
	const GeometricField<Type, PatchField, GeoMesh>& gf1
)
{
	sqr(gf.internalField(), gf1.internalField());
	sqr(gf.boundaryField(), gf1.boundaryField());
}

template<class Type, template<class> class PatchField, class GeoMesh>
tmp
<
	GeometricField
	<
		typename outerProduct<Type, Type>::type,
		PatchField,
		GeoMesh
	>
>
sqr(const GeometricField<Type, PatchField, GeoMesh>& gf)
{
	typedef typename outerProduct<Type, Type>::type outerProductType;

	tmp<GeometricField<outerProductType, PatchField, GeoMesh> > tSqr
	(
		new GeometricField<outerProductType, PatchField, GeoMesh>
		(
			IOobject
			(
				"sqr(" + gf.name() + ')',
				gf.instance(),
				gf.db(),
				IOobject::NO_READ,
				IOobject::NO_WRITE
			),
			gf.mesh(),
			sqr(gf.dimensions())
		)
	);

	sqr(tSqr(), gf);

	return tSqr;
}

template<class Type, template<class> class PatchField, class GeoMesh>
tmp
<
	GeometricField
	<
		typename outerProduct<Type, Type>::type,
		PatchField,
		GeoMesh
	>
>
sqr(const tmp<GeometricField<Type, PatchField, GeoMesh> >& tgf)
{
	typedef typename outerProduct<Type, Type>::type outerProductType;

	const GeometricField<Type, PatchField, GeoMesh>& gf = tgf();

	tmp<GeometricField<outerProductType, PatchField, GeoMesh> > tSqr
	(
		new GeometricField<outerProductType, PatchField, GeoMesh>
		(
			IOobject
			(
				"sqr(" + gf.name() + ')',
				gf.instance(),
				gf.db(),
				IOobject::NO_READ,
				IOobject::NO_WRITE
			),
			gf.mesh(),
			sqr(gf.dimensions())
		)
	);

	sqr(tSqr(), gf);

	tgf.clear();

	return tSqr;
}


template<class Type, template<class> class PatchField, class GeoMesh>
void magSqr
(
	GeometricField<scalar, PatchField, GeoMesh>& gsf,
	const GeometricField<Type, PatchField, GeoMesh>& gf
)
{
	magSqr(gsf.internalField(), gf.internalField());
	magSqr(gsf.boundaryField(), gf.boundaryField());
}

template<class Type, template<class> class PatchField, class GeoMesh>
tmp<GeometricField<scalar, PatchField, GeoMesh> > magSqr
(
	const GeometricField<Type, PatchField, GeoMesh>& gf
)
{
	tmp<GeometricField<scalar, PatchField, GeoMesh> > tMagSqr
	(
		new GeometricField<scalar, PatchField, GeoMesh>
		(
			IOobject
			(
				"magSqr(" + gf.name() + ')',
				gf.instance(),
				gf.db(),
				IOobject::NO_READ,
				IOobject::NO_WRITE
			),
			gf.mesh(),
			sqr(gf.dimensions())
		)
	);

	magSqr(tMagSqr(), gf);

	return tMagSqr;
}

template<class Type, template<class> class PatchField, class GeoMesh>
tmp<GeometricField<scalar, PatchField, GeoMesh> > magSqr
(
	const tmp<GeometricField<Type, PatchField, GeoMesh> >& tgf
)
{
	const GeometricField<Type, PatchField, GeoMesh>& gf = tgf();

	tmp<GeometricField<scalar, PatchField, GeoMesh> > tMagSqr
	(
		new GeometricField<scalar, PatchField, GeoMesh>
		(
			IOobject
			(
				"magSqr(" + gf.name() + ')',
				gf.instance(),
				gf.db(),
				IOobject::NO_READ,
				IOobject::NO_WRITE
			),
			gf.mesh(),
			sqr(gf.dimensions())
		)
	);

	magSqr(tMagSqr(), gf);

	tgf.clear();

	return tMagSqr;
}


template<class Type, template<class> class PatchField, class GeoMesh>
void mag
(
	GeometricField<scalar, PatchField, GeoMesh>& gsf,
	const GeometricField<Type, PatchField, GeoMesh>& gf
)
{
	mag(gsf.internalField(), gf.internalField());
	mag(gsf.boundaryField(), gf.boundaryField());
}

template<class Type, template<class> class PatchField, class GeoMesh>
tmp<GeometricField<scalar, PatchField, GeoMesh> > mag
(
	const GeometricField<Type, PatchField, GeoMesh>& gf
)
{
	tmp<GeometricField<scalar, PatchField, GeoMesh> > tMag
	(
		new GeometricField<scalar, PatchField, GeoMesh>
		(
			IOobject
			(
				"mag(" + gf.name() + ')',
				gf.instance(),
				gf.db(),
				IOobject::NO_READ,
				IOobject::NO_WRITE
			),
			gf.mesh(),
			gf.dimensions()
		)
	);

	mag(tMag(), gf);

	return tMag;
}

template<class Type, template<class> class PatchField, class GeoMesh>
tmp<GeometricField<scalar, PatchField, GeoMesh> > mag
(
	const tmp<GeometricField<Type, PatchField, GeoMesh> >& tgf
)
{
	const GeometricField<Type, PatchField, GeoMesh>& gf = tgf();

	tmp<GeometricField<scalar, PatchField, GeoMesh> > tMag
	(
		new GeometricField<scalar, PatchField, GeoMesh>
		(
			IOobject
			(
				"mag(" + gf.name() + ')',
				gf.instance(),
				gf.db(),
				IOobject::NO_READ,
				IOobject::NO_WRITE
			),
			gf.mesh(),
			gf.dimensions()
		)
	);

	mag(tMag(), gf);

	tgf.clear();

	return tMag;
}


template<class Type, template<class> class PatchField, class GeoMesh>
void cmptAv
(
	GeometricField
	<
		typename GeometricField<Type, PatchField, GeoMesh>::cmptType,
		PatchField,
		GeoMesh
	>& gcf,
	const GeometricField<Type, PatchField, GeoMesh>& gf
)
{
	cmptAv(gcf.internalField(), gf.internalField());
	cmptAv(gcf.boundaryField(), gf.boundaryField());
}

template<class Type, template<class> class PatchField, class GeoMesh>
tmp
<
	GeometricField
	<
		typename GeometricField<Type, PatchField, GeoMesh>::cmptType,
		PatchField,
		GeoMesh
	>
>
cmptAv(const GeometricField<Type, PatchField, GeoMesh>& gf)
{
	typedef typename GeometricField<Type, PatchField, GeoMesh>::cmptType
		cmptType;

	tmp<GeometricField<cmptType, PatchField, GeoMesh> > CmptAv
	(
		new GeometricField<scalar, PatchField, GeoMesh>
		(
			IOobject
			(
				"cmptAv(" + gf.name() + ')',
				gf.instance(),
				gf.db(),
				IOobject::NO_READ,
				IOobject::NO_WRITE
			),
			gf.mesh(),
			gf.dimensions()
		)
	);

	cmptAv(CmptAv(), gf);

	return CmptAv;
}

template<class Type, template<class> class PatchField, class GeoMesh>
tmp
<
	GeometricField
	<
		typename GeometricField<Type, PatchField, GeoMesh>::cmptType,
		PatchField,
		GeoMesh
	>
>
cmptAv(const tmp<GeometricField<Type, PatchField, GeoMesh> >& tgf)
{
	typedef typename GeometricField<Type, PatchField, GeoMesh>::cmptType
		cmptType;

	const GeometricField<Type, PatchField, GeoMesh>& gf = tgf();

	tmp<GeometricField<cmptType, PatchField, GeoMesh> > CmptAv
	(
		new GeometricField<scalar, PatchField, GeoMesh>
		(
			IOobject
			(
				"cmptAv(" + gf.name() + ')',
				gf.instance(),
				gf.db(),
				IOobject::NO_READ,
				IOobject::NO_WRITE
			),
			gf.mesh(),
			gf.dimensions()
		)
	);

	cmptAv(CmptAv(), gf);

	tgf.clear();

	return CmptAv;
}


#define UNARY_REDUCTION_FUNCTION_WITH_BOUNDARY(returnType, func, gFunc)       \
					                                                          \
template<class Type, template<class> class PatchField, class GeoMesh>         \
dimensioned<returnType> func                                                  \
(                                                                             \
	const GeometricField<Type, PatchField, GeoMesh>& gf                       \
)                                                                             \
{                                                                             \
	return dimensioned<Type>                                                  \
	(                                                                         \
		#func "(" + gf.name() + ')',                                          \
		gf.dimensions(),                                                      \
		Foam::func(gFunc(gf.internalField()), gFunc(gf.boundaryField()))      \
	);                                                                        \
}                                                                             \
					                                                          \
template<class Type, template<class> class PatchField, class GeoMesh>         \
dimensioned<returnType> func                                                  \
(                                                                             \
	const tmp<GeometricField<Type, PatchField, GeoMesh> >& tgf1               \
)                                                                             \
{                                                                             \
	dimensioned<returnType> res = func(tgf1());                               \
	tgf1.clear();                                                             \
	return res;                                                               \
}

UNARY_REDUCTION_FUNCTION_WITH_BOUNDARY(Type, max, gMax)
UNARY_REDUCTION_FUNCTION_WITH_BOUNDARY(Type, min, gMin)

#undef UNARY_REDUCTION_FUNCTION_WITH_BOUNDARY


#define UNARY_REDUCTION_FUNCTION(returnType, func, gFunc)                     \
					                                                          \
template<class Type, template<class> class PatchField, class GeoMesh>         \
dimensioned<returnType> func                                                  \
(                                                                             \
	const GeometricField<Type, PatchField, GeoMesh>& gf                       \
)                                                                             \
{                                                                             \
	return dimensioned<Type>                                                  \
	(                                                                         \
		#func "(" + gf.name() + ')',                                          \
		gf.dimensions(),                                                      \
		gFunc(gf.internalField())                                             \
	);                                                                        \
}                                                                             \
					                                                          \
template<class Type, template<class> class PatchField, class GeoMesh>         \
dimensioned<returnType> func                                                  \
(                                                                             \
	const tmp<GeometricField<Type, PatchField, GeoMesh> >& tgf1               \
)                                                                             \
{                                                                             \
	dimensioned<returnType> res = func(tgf1());                               \
	tgf1.clear();                                                             \
	return res;                                                               \
}

UNARY_REDUCTION_FUNCTION(Type, sum, gSum)
UNARY_REDUCTION_FUNCTION(scalar, sumMag, gSumMag)
UNARY_REDUCTION_FUNCTION(Type, average, gAverage)

#undef UNARY_REDUCTION_FUNCTION


BINARY_FUNCTION(Type, Type, Type, max)
BINARY_FUNCTION(Type, Type, Type, min)
BINARY_FUNCTION(Type, Type, Type, cmptMultiply)
BINARY_FUNCTION(Type, Type, Type, cmptDivide)

BINARY_TYPE_FUNCTION(Type, Type, Type, max)
BINARY_TYPE_FUNCTION(Type, Type, Type, min)
BINARY_TYPE_FUNCTION(Type, Type, Type, cmptMultiply)
BINARY_TYPE_FUNCTION(Type, Type, Type, cmptDivide)


// * * * * * * * * * * * * * * * Global operators  * * * * * * * * * * * * * //

UNARY_OPERATOR(Type, Type, -, negate, transform)

#ifndef __INTEL_COMPILER
BINARY_OPERATOR(Type, Type, scalar, *, '*', multiply)
BINARY_OPERATOR(Type, scalar, Type, *, '*', multiply)
#endif
BINARY_OPERATOR(Type, Type, scalar, /, '|', divide)

BINARY_TYPE_OPERATOR_SF(Type, scalar, Type, *, '*', multiply)
BINARY_TYPE_OPERATOR_FS(Type, Type, scalar, *, '*', multiply)

BINARY_TYPE_OPERATOR_FS(Type, Type, scalar, /, '|', divide)


// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * //

#define PRODUCT_OPERATOR(product, op, opFunc)                                 \
					                                                          \
template                                                                      \
<class Type1, class Type2, template<class> class PatchField, class GeoMesh>   \
void opFunc                                                                   \
(                                                                             \
	GeometricField                                                            \
	<typename product<Type1, Type2>::type, PatchField, GeoMesh>& gf,          \
	const GeometricField<Type1, PatchField, GeoMesh>& gf1,                    \
	const GeometricField<Type2, PatchField, GeoMesh>& gf2                     \
)                                                                             \
{                                                                             \
	Foam::opFunc(gf.internalField(), gf1.internalField(), gf2.internalField());\
	Foam::opFunc(gf.boundaryField(), gf1.boundaryField(), gf2.boundaryField());\
}                                                                             \
					                                                          \
template                                                                      \
<class Type1, class Type2, template<class> class PatchField, class GeoMesh>   \
tmp                                                                           \
<                                                                             \
	GeometricField<typename product<Type1, Type2>::type, PatchField, GeoMesh> \
>                                                                             \
operator op                                                                   \
(                                                                             \
	const GeometricField<Type1, PatchField, GeoMesh>& gf1,                    \
	const GeometricField<Type2, PatchField, GeoMesh>& gf2                     \
)                                                                             \
{                                                                             \
	typedef typename product<Type1, Type2>::type productType;                 \
	tmp<GeometricField<productType, PatchField, GeoMesh> > tRes               \
	(                                                                         \
		new GeometricField<productType, PatchField, GeoMesh>                  \
		(                                                                     \
			IOobject                                                          \
			(                                                                 \
				'(' + gf1.name() + #op + gf2.name() + ')',                    \
				gf1.instance(),                                               \
				gf1.db(),                                                     \
				IOobject::NO_READ,                                            \
				IOobject::NO_WRITE                                            \
			),                                                                \
			gf1.mesh(),                                                       \
			gf1.dimensions() op gf2.dimensions()                              \
		)                                                                     \
	);                                                                        \
					                                                          \
	Foam::opFunc(tRes(), gf1, gf2);                                           \
					                                                          \
	return tRes;                                                              \
}                                                                             \
					                                                          \
template                                                                      \
<class Type1, class Type2, template<class> class PatchField, class GeoMesh>   \
tmp                                                                           \
<                                                                             \
	GeometricField<typename product<Type1, Type2>::type, PatchField, GeoMesh> \
>                                                                             \
operator op                                                                   \
(                                                                             \
	const GeometricField<Type1, PatchField, GeoMesh>& gf1,                    \
	const tmp<GeometricField<Type2, PatchField, GeoMesh> >& tgf2              \
)                                                                             \
{                                                                             \
	typedef typename product<Type1, Type2>::type productType;                 \
					                                                          \
	const GeometricField<Type2, PatchField, GeoMesh>& gf2 = tgf2();           \
					                                                          \
	tmp<GeometricField<productType, PatchField, GeoMesh> > tRes =             \
		reuseTmpGeometricField<productType, Type2, PatchField, GeoMesh>::New  \
		(                                                                     \
			tgf2,                                                             \
			'(' + gf1.name() + #op + gf2.name() + ')',                        \
			gf1.dimensions() op gf2.dimensions()                              \
		);                                                                    \
					                                                          \
	Foam::opFunc(tRes(), gf1, gf2);                                           \
					                                                          \
	reuseTmpGeometricField<productType, Type2, PatchField, GeoMesh>           \
	::clear(tgf2);                                                            \
					                                                          \
	return tRes;                                                              \
}                                                                             \
					                                                          \
template                                                                      \
<class Type1, class Type2, template<class> class PatchField, class GeoMesh>   \
tmp                                                                           \
<                                                                             \
	GeometricField<typename product<Type1, Type2>::type, PatchField, GeoMesh> \
>                                                                             \
operator op                                                                   \
(                                                                             \
	const tmp<GeometricField<Type1, PatchField, GeoMesh> >& tgf1,             \
	const GeometricField<Type2, PatchField, GeoMesh>& gf2                     \
)                                                                             \
{                                                                             \
	typedef typename product<Type1, Type2>::type productType;                 \
					                                                          \
	const GeometricField<Type1, PatchField, GeoMesh>& gf1 = tgf1();           \
					                                                          \
	tmp<GeometricField<productType, PatchField, GeoMesh> > tRes =             \
		reuseTmpGeometricField<productType, Type1, PatchField, GeoMesh>::New  \
		(                                                                     \
			tgf1,                                                             \
			'(' + gf1.name() + #op + gf2.name() + ')',                        \
			gf1.dimensions() op gf2.dimensions()                              \
		);                                                                    \
					                                                          \
	Foam::opFunc(tRes(), gf1, gf2);                                           \
					                                                          \
	reuseTmpGeometricField<productType, Type1, PatchField, GeoMesh>           \
	::clear(tgf1);                                                            \
					                                                          \
	return tRes;                                                              \
}                                                                             \
					                                                          \
template                                                                      \
<class Type1, class Type2, template<class> class PatchField, class GeoMesh>   \
tmp                                                                           \
<                                                                             \
	GeometricField<typename product<Type1, Type2>::type, PatchField, GeoMesh> \
>                                                                             \
operator op                                                                   \
(                                                                             \
	const tmp<GeometricField<Type1, PatchField, GeoMesh> >& tgf1,             \
	const tmp<GeometricField<Type2, PatchField, GeoMesh> >& tgf2              \
)                                                                             \
{                                                                             \
	typedef typename product<Type1, Type2>::type productType;                 \
					                                                          \
	const GeometricField<Type1, PatchField, GeoMesh>& gf1 = tgf1();           \
	const GeometricField<Type2, PatchField, GeoMesh>& gf2 = tgf2();           \
					                                                          \
	tmp<GeometricField<productType, PatchField, GeoMesh> > tRes =             \
		reuseTmpTmpGeometricField                                             \
		<productType, Type1, Type1, Type2, PatchField, GeoMesh>::New          \
		(                                                                     \
			tgf1,                                                             \
			tgf2,                                                             \
			'(' + gf1.name() + #op + gf2.name() + ')',                        \
			gf1.dimensions() op gf2.dimensions()                              \
		);                                                                    \
					                                                          \
	Foam::opFunc(tRes(), gf1, gf2);                                           \
					                                                          \
	reuseTmpTmpGeometricField                                                 \
		<productType, Type1, Type1, Type2, PatchField, GeoMesh>               \
	::clear(tgf1, tgf2);                                                      \
					                                                          \
	return tRes;                                                              \
}                                                                             \
					                                                          \
template                                                                      \
<class Form, class Type, template<class> class PatchField, class GeoMesh>     \
void opFunc                                                                   \
(                                                                             \
	GeometricField                                                            \
	<typename product<Type, Form>::type, PatchField, GeoMesh>& gf,            \
	const GeometricField<Type, PatchField, GeoMesh>& gf1,                     \
	const dimensioned<Form>& dvs                                              \
)                                                                             \
{                                                                             \
	Foam::opFunc(gf.internalField(), gf1.internalField(), dvs.value());       \
	Foam::opFunc(gf.boundaryField(), gf1.boundaryField(), dvs.value());       \
}                                                                             \
					                                                          \
template                                                                      \
<class Form, class Type, template<class> class PatchField, class GeoMesh>     \
tmp<GeometricField<typename product<Type, Form>::type, PatchField, GeoMesh> > \
operator op                                                                   \
(                                                                             \
	const GeometricField<Type, PatchField, GeoMesh>& gf1,                     \
	const dimensioned<Form>& dvs                                              \
)                                                                             \
{                                                                             \
	typedef typename product<Type, Form>::type productType;                   \
					                                                          \
	tmp<GeometricField<productType, PatchField, GeoMesh> > tRes               \
	(                                                                         \
		new GeometricField<productType, PatchField, GeoMesh>                  \
		(                                                                     \
			IOobject                                                          \
			(                                                                 \
				'(' + gf1.name() + #op + dvs.name() + ')',                    \
				gf1.instance(),                                               \
				gf1.db(),                                                     \
				IOobject::NO_READ,                                            \
				IOobject::NO_WRITE                                            \
			),                                                                \
			gf1.mesh(),                                                       \
			gf1.dimensions() op dvs.dimensions()                              \
		)                                                                     \
	);                                                                        \
					                                                          \
	Foam::opFunc(tRes(), gf1, dvs);                                           \
					                                                          \
	return tRes;                                                              \
}                                                                             \
					                                                          \
template                                                                      \
<                                                                             \
	class Form,                                                               \
	class Cmpt,                                                               \
	int nCmpt,                                                                \
	class Type, template<class> class PatchField,                             \
	class GeoMesh                                                             \
>                                                                             \
tmp<GeometricField<typename product<Form, Type>::type, PatchField, GeoMesh> > \
operator op                                                                   \
(                                                                             \
	const GeometricField<Type, PatchField, GeoMesh>& gf1,                     \
	const VectorSpace<Form,Cmpt,nCmpt>& vs                                    \
)                                                                             \
{                                                                             \
	return gf1 op dimensioned<Form>(static_cast<const Form&>(vs));            \
}                                                                             \
					                                                          \
					                                                          \
template                                                                      \
<class Form, class Type, template<class> class PatchField, class GeoMesh>     \
tmp<GeometricField<typename product<Type, Form>::type, PatchField, GeoMesh> > \
operator op                                                                   \
(                                                                             \
	const tmp<GeometricField<Type, PatchField, GeoMesh> >& tgf1,              \
	const dimensioned<Form>& dvs                                              \
)                                                                             \
{                                                                             \
	typedef typename product<Type, Form>::type productType;                   \
					                                                          \
	const GeometricField<Type, PatchField, GeoMesh>& gf1 = tgf1();            \
					                                                          \
	tmp<GeometricField<productType, PatchField, GeoMesh> > tRes =             \
		reuseTmpGeometricField<productType, Type, PatchField, GeoMesh>::New   \
		(                                                                     \
			tgf1,                                                             \
			'(' + gf1.name() + #op + dvs.name() + ')',                        \
			gf1.dimensions() op dvs.dimensions()                              \
		);                                                                    \
					                                                          \
	Foam::opFunc(tRes(), gf1, dvs);                                           \
					                                                          \
	reuseTmpGeometricField<productType, Type, PatchField, GeoMesh>            \
	::clear(tgf1);                                                            \
					                                                          \
	return tRes;                                                              \
}                                                                             \
					                                                          \
template                                                                      \
<                                                                             \
	class Form,                                                               \
	class Cmpt,                                                               \
	int nCmpt,                                                                \
	class Type, template<class> class PatchField,                             \
	class GeoMesh                                                             \
>                                                                             \
tmp<GeometricField<typename product<Form, Type>::type, PatchField, GeoMesh> > \
operator op                                                                   \
(                                                                             \
	const tmp<GeometricField<Type, PatchField, GeoMesh> >& tgf1,              \
	const VectorSpace<Form,Cmpt,nCmpt>& vs                                    \
)                                                                             \
{                                                                             \
	return tgf1 op dimensioned<Form>(static_cast<const Form&>(vs));           \
}                                                                             \
					                                                          \
					                                                          \
template                                                                      \
<class Form, class Type, template<class> class PatchField, class GeoMesh>     \
void opFunc                                                                   \
(                                                                             \
	GeometricField                                                            \
	<typename product<Form, Type>::type, PatchField, GeoMesh>& gf,            \
	const dimensioned<Form>& dvs,                                             \
	const GeometricField<Type, PatchField, GeoMesh>& gf1                      \
)                                                                             \
{                                                                             \
	Foam::opFunc(gf.internalField(), dvs.value(), gf1.internalField());       \
	Foam::opFunc(gf.boundaryField(), dvs.value(), gf1.boundaryField());       \
}                                                                             \
					                                                          \
template                                                                      \
<class Form, class Type, template<class> class PatchField, class GeoMesh>     \
tmp<GeometricField<typename product<Form, Type>::type, PatchField, GeoMesh> > \
operator op                                                                   \
(                                                                             \
	const dimensioned<Form>& dvs,                                             \
	const GeometricField<Type, PatchField, GeoMesh>& gf1                      \
)                                                                             \
{                                                                             \
	typedef typename product<Form, Type>::type productType;                   \
	tmp<GeometricField<productType, PatchField, GeoMesh> > tRes               \
	(                                                                         \
		new GeometricField<productType, PatchField, GeoMesh>                  \
		(                                                                     \
			IOobject                                                          \
			(                                                                 \
				'(' + dvs.name() + #op + gf1.name() + ')',                    \
				gf1.instance(),                                               \
				gf1.db(),                                                     \
				IOobject::NO_READ,                                            \
				IOobject::NO_WRITE                                            \
			),                                                                \
			gf1.mesh(),                                                       \
			dvs.dimensions() op gf1.dimensions()                              \
		)                                                                     \
	);                                                                        \
					                                                          \
	Foam::opFunc(tRes(), dvs, gf1);                                           \
					                                                          \
	return tRes;                                                              \
}                                                                             \
					                                                          \
template                                                                      \
<                                                                             \
	class Form,                                                               \
	class Cmpt,                                                               \
	int nCmpt,                                                                \
	class Type, template<class> class PatchField,                             \
	class GeoMesh                                                             \
>                                                                             \
tmp<GeometricField<typename product<Form, Type>::type, PatchField, GeoMesh> > \
operator op                                                                   \
(                                                                             \
	const VectorSpace<Form,Cmpt,nCmpt>& vs,                                   \
	const GeometricField<Type, PatchField, GeoMesh>& gf1                      \
)                                                                             \
{                                                                             \
	return dimensioned<Form>(static_cast<const Form&>(vs)) op gf1;            \
}                                                                             \
					                                                          \
template                                                                      \
<class Form, class Type, template<class> class PatchField, class GeoMesh>     \
tmp<GeometricField<typename product<Form, Type>::type, PatchField, GeoMesh> > \
operator op                                                                   \
(                                                                             \
	const dimensioned<Form>& dvs,                                             \
	const tmp<GeometricField<Type, PatchField, GeoMesh> >& tgf1               \
)                                                                             \
{                                                                             \
	typedef typename product<Form, Type>::type productType;                   \
					                                                          \
	const GeometricField<Type, PatchField, GeoMesh>& gf1 = tgf1();            \
					                                                          \
	tmp<GeometricField<productType, PatchField, GeoMesh> > tRes =             \
		reuseTmpGeometricField<productType, Type, PatchField, GeoMesh>::New   \
		(                                                                     \
			tgf1,                                                             \
			'(' + dvs.name() + #op + gf1.name() + ')',                        \
			dvs.dimensions() op gf1.dimensions()                              \
		);                                                                    \
					                                                          \
	Foam::opFunc(tRes(), dvs, gf1);                                           \
					                                                          \
	reuseTmpGeometricField<productType, Type, PatchField, GeoMesh>            \
	::clear(tgf1);                                                            \
					                                                          \
	return tRes;                                                              \
}                                                                             \
					                                                          \
template                                                                      \
<                                                                             \
	class Form,                                                               \
	class Cmpt,                                                               \
	int nCmpt,                                                                \
	class Type, template<class> class PatchField,                             \
	class GeoMesh                                                             \
>                                                                             \
tmp<GeometricField<typename product<Form, Type>::type, PatchField, GeoMesh> > \
operator op                                                                   \
(                                                                             \
	const VectorSpace<Form,Cmpt,nCmpt>& vs,                                   \
	const tmp<GeometricField<Type, PatchField, GeoMesh> >& tgf1               \
)                                                                             \
{                                                                             \
	return dimensioned<Form>(static_cast<const Form&>(vs)) op tgf1;           \
}

PRODUCT_OPERATOR(typeOfSum, +, add)
PRODUCT_OPERATOR(typeOfSum, -, subtract)

PRODUCT_OPERATOR(outerProduct, *, outer)
PRODUCT_OPERATOR(crossProduct, ^, cross)
PRODUCT_OPERATOR(innerProduct, &, dot)
PRODUCT_OPERATOR(scalarProduct, &&, dotdot)

#undef PRODUCT_OPERATOR


// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * //

} // End namespace Foam

// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * //

#include "undefFieldFunctionsM.H"

// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * //
